<template>
    <!--判断scopedSlots.default可知道是否存在子元素-->
    <el-table-column
      v-if="$scopedSlots.default"
      v-bind="$attrs"
      :key="$attrs.label"
      :class-name="className"
      :min-width="minWidth"
    >
      <template slot-scope="scope">
        <!--原element组件使用了作用域插槽, 这里仿照-->
        <slot v-bind="scope"></slot>
      </template>
    </el-table-column>
    <!--使用 slot 自定义 header-->
    <el-table-column
      v-else-if="$scopedSlots.header"
      v-bind="$attrs"
      :key="$attrs.label"
      :class-name="className"
      :min-width="minWidth"
    >
      <template slot="header" slot-scope="scope">
        <slot name="header" v-bind="scope"></slot>
      </template>
    </el-table-column>
    <!--默认情况使用原始 el-table-column-->
    <el-table-column
      v-else
      v-bind="$attrs"
      :key="$attrs.label"
      :class-name="className"
      :min-width="minWidth"
    ></el-table-column>
  </template>
  
  <script>
  import $consts from './constants'
  export default {
    name: 'af-table-column',
    computed: {
      // table数据
      values () {
        const data = this.$parent.data || []
        return data.map(item => item[this.$attrs.prop])
      },
      // 是否自适应列宽, 默认是
      isFit () {
        return (this.$parent.$attrs.autoFit === undefined && this.$attrs.fit === undefined) || (this.$parent.$attrs.autoFit === false && this.$attrs.fit !== undefined)
      },
      // 为存在scope的添加className
      className () {
        const parentClass = this.$attrs['class-name'] || ''
        const classPre = this.$attrs.prop || `encode-${this.transChar(this.$attrs.label)}` // 有的列因为有slotScope而不给prop
        return classPre ? `${parentClass} ${classPre}-column` : ''
      },
      // 列最小宽度
      minWidth () {
        if (!this.$attrs.label) return undefined
        if (!this.isFit) return undefined
        const maxOne = Math.max(this.minLength, this.$attrs.label.length * this.fontRate.CHAR_RATE) * this.fontSize + 20
        return this.$attrs.width || Math.max(maxOne, this.getComputedWidth)
      },
      // 字体大小
      fontSize () {
        return this.$attrs.fontSize || (this.$ELEMENT || {}).fontSize || $consts.fontSize
      },
      // 字体比例
      fontRate () {
        return {
          ...$consts.fontRate, // 默认值
          ...(this.$ELEMENT || {}).fontRate || {}, // 根注册参数 (Vue.use 时的第二个参数)
          ... this.$attrs.fontRate || {}, // 父组件属性
        }
      }
    },
    watch: {
      values: {
        immediate: true,
        handler (val) {
          this.isFit !== false && this.$nextTick(() => {
            // 详情中的列表需要在nextTick才能生效
            if (this.$scopedSlots.default) {
              // 存在自定义节点
              setTimeout(() => {
                // 首次获取不到子节点, setTimeout才行
                // 可能存在贴边列, 需要使用包含 fixed 的类名
                const bodyWrapper = this.$attrs.fixed ? document.querySelector(`.el-table__fixed${this.$attrs.fixed === 'right' ? `-${this.$attrs.fixed}` : ''}`).querySelector('.el-table__fixed-body-wrapper') : document.querySelector('.el-table__body-wrapper')
                const nodes = bodyWrapper.querySelectorAll(`.${this.$attrs.prop || `encode-${this.transChar(this.$attrs.label)}`}-column`)
                if (nodes.length) {
                  // 有可能会来不及获取nodes, 就切换菜单进入下一页了, 再研究吧
                  const target = []
                  const getComputedWidths = []
                  Array.from(nodes).map(item => {
                    target.push(item.innerText)
                    // 有可能存在自定义列内容超出容器, 取 scrollWidth 才行
                    getComputedWidths.push(item.querySelector('.cell').scrollWidth)
                  })
                  this.getComputedWidth = Math.max(...getComputedWidths)
                  this.$set(this, 'minLength', this.getMaxLength(target))
                }
              }, 0)
            } else {
              this.$set(this, 'minLength', this.getMaxLength(val))
            }
          })
        }
      }
    },
    data () {
      return {
        minLength: 5, // 初始也不要写成0, 避免未及时设置宽度太丑
        getComputedWidth: 0, // 自定义列中获取元素计算的宽度
      }
    },
    methods: {
      // 获取数组中元素按字体比例最长长度
      getMaxLength (arr) {
        return arr.reduce((length, item) => {
          if (item) {
            const str = item.toString()
            const char = str.match(/[\u2E80-\u9FFF]/g)
            const charLength = char ? char.length : 0
            const num = str.match(/\d|\./g)
            const numLength = num ? num.length : 0
            const otherLength = str.length - charLength - numLength
            let newLength = charLength * this.fontRate.CHAR_RATE + numLength * this.fontRate.NUM_RATE + otherLength * this.fontRate.OTHER_RATE
            if (str.includes('\n')) newLength = this.getMaxLength(str.split('\n'))
            if (length < newLength) {
              length = newLength
            }
          }
          return length
        }, 0)
      },
      // 转换汉字为class支持的字母
      transChar (char) {
        return encodeURIComponent(char).replace(/[^a-zA-z]/g, 'eUC')
      }
    },
  }
  </script>